package com.atguigu.interview.study.thread;

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 * @author gcq
 * @Create 2020-11-26
 * 题目：Synchronized和Lock有什么区别？你举例说说？
 * 1、原始构成
 *      Synchronized是关键字属于JVM层面
 *      monitorentor(底层是通过Moniter对象完成，其实wait/notify等方法也依赖于monitor对象只有在同步快或方法中才能调wait/notify等方法)
 *      monitorexit
 *   Lock是具体类(java.util.concurrent.Locks.lock)是api层面的锁
 * 2、使用方法
 *      synchronized 不需用户手动释放锁，当Synchronized代码执行完之后系统会自动让线程释放对锁的占用
 *      ReentrantLock需要用户去手动释放锁若没有主动释放锁，就有可能导致出现死锁现象
 *      需要lock()和unlock()配合try/finally语句块来完成
 * 3、等待是否中断
 *      Synchronized 不可中断，除非抛出异常或正常运行完成
 *      ReentrantLock 可中断，1、设置超时方法 tryLock(long timeout, TimeUnit unit)
 *                           2、lockInterruptibly()放代码块中，调用interrupt()可中断
 * 4、加锁是否公平
 *      Synchronized 非公平锁
 *      ReentrantLock两者都可以，默认非公平锁，构造方法可以传入boolean的值，true为公平锁，false为非公平锁，默认flase
 * 5、锁绑定多个Condition
 *      Synchronized 没有
 *      ReentrantLock用来实现分组唤醒需要唤醒的线程，可以精确唤醒，而不是像Synchronized要么随机唤醒一个线程要么唤醒全部线程
 *
 * =================================================================================================================
 * =================================================================================================================
 *
 * 题目: 多线程之间按照顺序调用，实现A->B->C三个线程启动，要求如下
 * AA打印5次，BB打印10次，CC打印15次
 * 紧接着
 * AA打印5次，BB打印10次，CC打印15次
 * 来10轮
 */

/***
 *
 */
class ShareResource
{
    /**
     * A1 B2 C3
     */
    private int number = 1;

    private Lock lock = new ReentrantLock();
    private Condition c1 = lock.newCondition();
    private Condition c2 = lock.newCondition();
    private Condition c3 = lock.newCondition();


    public void print5() {
        lock.lock();
        try {
            // 1、判断
            while(number != 1) {
                // c1等待
                c1.await();
            }
            // 2、干活
            for(int i = 1; i <= 5; i++) {
                System.out.println(Thread.currentThread().getName()+"\t" + i);
            }
            // 3、通知
            number = 2;
            // 通知c2
            c2.signal();
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }

    public void print10() {
        lock.lock();
        try {
            // 1、判断
            while(number != 2) {
                // c1等待
                c2.await();
            }
            // 2、干活
            for(int i = 1; i <= 10; i++) {
                System.out.println(Thread.currentThread().getName()+"\t" + i);
            }
            // 3、通知
            number = 3;
            // 通知c2
            c3.signal();
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }


    public void print15() {
        lock.lock();
        try {
            // 1、判断
            while(number != 3) {
                // c1等待
                c3.await();
            }
            // 2、干活
            for(int i = 1; i <= 15; i++) {
                System.out.println(Thread.currentThread().getName()+"\t" + i);
            }
            // 3、通知
            number = 1;
            // 通知c2
            c1.signal();
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }

}
public class SyncAndReentrantLockDemo {
    public static void main(String[] args) {
        ShareResource shareResource = new ShareResource();

        new Thread(() -> {
            for(int i = 1; i <= 10; i++)
            {
                shareResource.print5();
            }
        },"A").start();

        new Thread(() -> {
            for(int i = 1; i <= 10; i++)
            {
                shareResource.print10();
            }
        },"B").start();

        new Thread(() -> {
            for(int i = 1; i <= 10; i++)
            {
                shareResource.print15();
            }
        },"C").start();
    }
}